//序列化是将一个数据结构或者对象转换为连续的比特位的操作，进而可以将转换后的数据存储在一个文件或者内存中，同时也可以通过网络传输到另一个计算机环境，采取相反方
//式重构得到原数据。 
//
// 请设计一个算法来实现二叉树的序列化与反序列化。这里不限定你的序列 / 反序列化算法执行逻辑，你只需要保证一个二叉树可以被序列化为一个字符串并且将这个字符串
//反序列化为原始的树结构。 
//
// 提示: 输入输出格式与 LeetCode 目前使用的方式一致，详情请参阅 LeetCode 序列化二叉树的格式。你并非必须采取这种方式，你也可以采用其他的
//方法解决这个问题。 
//
// 
//
// 示例 1： 
//
// 
//输入：root = [1,2,3,null,null,4,5]
//输出：[1,2,3,null,null,4,5]
// 
//
// 示例 2： 
//
// 
//输入：root = []
//输出：[]
// 
//
// 示例 3： 
//
// 
//输入：root = [1]
//输出：[1]
// 
//
// 示例 4： 
//
// 
//输入：root = [1,2]
//输出：[1,2]
// 
//
// 
//
// 提示： 
//
// 
// 树中结点数在范围 [0, 10⁴] 内 
// -1000 <= Node.val <= 1000 
// 
// Related Topics 树 深度优先搜索 广度优先搜索 设计 字符串 二叉树 👍 907 👎 0

package leetcode.editor.cn;

import java.util.*;

class SerializeAndDeserializeBinaryTree {
    public static void main(String[] args) {
        Codec solution = new SerializeAndDeserializeBinaryTree().new Codec();
    }

    //leetcode submit region begin(Prohibit modification and deletion)
    public class TreeNode {
        int val;
        TreeNode left;
        TreeNode right;

        TreeNode(int x) {
            val = x;
        }
    }

    public class Codec {

        // Encodes a tree to a single string.
        /*public String serialize(TreeNode root) {
            if (root == null) return "[]";
            Queue<TreeNode> queue = new LinkedList<>();
            queue.add(root);
            StringBuilder res = new StringBuilder().append("[");
            while (!queue.isEmpty()) {
                TreeNode node = queue.poll();
                if(node != null) {
                    res.append(node.val + ",");
                    queue.add(node.left);
                    queue.add(node.right);
                }
                else res.append("null,");
            }
            res.deleteCharAt(res.length() - 1);
            res.append("]");
            return res.toString();
        }

        // Decodes your encoded data to tree.
        public TreeNode deserialize(String data) {
            if (data.equals("[]")) return null;
            String[] split = data.substring(1, data.length() - 1).split(",");
            TreeNode root = new TreeNode(Integer.parseInt(split[0]));
            Queue<TreeNode> queue = new LinkedList<>();
            queue.add(root);
            int i = 1;
            while (!queue.isEmpty()) {
                TreeNode node = queue.poll();
                if (!split[i].equals("null")) {
                    node.left = new TreeNode(Integer.parseInt(split[i]));
                    queue.offer(node.left);
                }
                // 为null也要+1
                i++;

                if (!split[i].equals("null")) {
                    node.right = new TreeNode(Integer.parseInt(split[i]));
                    queue.offer(node.right);
                }
                i++;
            }

            return root;
        }*/

        // Encodes a tree to a single string.
        public String serialize(TreeNode root) {
            String rserialize = rserialize(root, "");
            return rserialize;
        }

        public String rserialize(TreeNode root, String str) {
            if (root == null) {
                str += "null,";
                return str;
            }

            str += root.val + ",";
            str = rserialize(root.left, str);
            str = rserialize(root.right, str);
            return str;
        }

        // Decodes your encoded data to tree.
        public TreeNode deserialize(String data) {
            String[] split = data.split(",");
            List<String> list = new ArrayList<String>(Arrays.asList(split));
            return rdeserialize(list);
        }

        public TreeNode rdeserialize(List<String> dataList) {
            // 递归到null停止，并删除
            if (dataList.get(0).equals("null")) {
                dataList.remove(0);
                return null;
            }

            TreeNode node = new TreeNode(Integer.parseInt(dataList.get(0)));
            // 注意生成节点之后删除元素
            dataList.remove(0);
            node.left = rdeserialize(dataList);
            node.right = rdeserialize(dataList);
            return node;
        }
    }

// Your Codec object will be instantiated and called as such:
// Codec ser = new Codec();
// Codec deser = new Codec();
// TreeNode ans = deser.deserialize(ser.serialize(root));
//leetcode submit region end(Prohibit modification and deletion)

}
